Widget简介

可以使用AppWidgetManager更新Widget中的数据,但这样最短也要半个小时才能更新一次,一般不用他更新,而是自己定义一个服务去更新Widget中的数据。

Widget的创建步骤

  1. 写一个类继承AppWidgetProvider,这个是一个广播接收者,所以要在清单文件中进行配置

     public class MyWidget extends AppWidgetProvider {
         @Override
         public void onEnabled(Context context) {
             //开启服务定期的更新界面.
             Intent intent = new Intent(context,UpdateWidgetService.class);
             context.startService(intent);
             super.onEnabled(context);
         }
    
         @Override
         public void onDisabled(Context context) {
             //关闭掉服务
             Intent intent = new Intent(context,UpdateWidgetService.class);
             context.stopService(intent);
             super.onDisabled(context);
         }
    
         @Override
         public void onUpdate(Context context, AppWidgetManager appWidgetManager,
                 int[] appWidgetIds) {      
    
             //Widget布局中定义的更新时间到了。检查下 服务是否还活着.
             if(!ServiceStatusUtil.isServiceRunning(context, "com.itheima.mobilesafe.service.UpdateWidgetService")){
                 Intent intent = new Intent(context,UpdateWidgetService.class);
                 context.startService(intent);
             }
             super.onUpdate(context, appWidgetManager, appWidgetIds);
         }
     }
    
  2. 在清单文件中进行配置,内容如下:

     <receiver android:name=".receiver.MyWidget" >
         <intent-filter>
             <action android:name="android.appwidget.action.APPWIDGET_UPDATE" />
         </intent-filter>
         <meta-data
             android:name="android.appwidget.provider"
             android:resource="@xml/example_appwidget_info" />//这里使用到了一个xml文件,所以要创建这个文件
     </receiver>
    
  3. 在res下面新建一个名为xml的文件件,然后新建example_appwidget_info.xml内容如下

     <?xml version="1.0" encoding="utf-8"?>
     <appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android"
         android:minWidth="294dp"
         android:minHeight="72dp"
         android:updatePeriodMillis="86400000" //指定更新的间隔时间,最小为半个小时,一般不用它更新,都是自己更新
         android:previewImage="@drawable/preview"//指定小控件的图标,如果不要这个选项就是程序的图标
         android:initialLayout="@layout/example_appwidget"//设置这个小控件的布局文件
         android:configure="com.example.android.ExampleAppWidgetConfigure" //有些复杂的Widget在点击的时候会去开启一个Activity,
         // 这时候就是通过这个参数来配置要开启的Activity,用不着就删除这一行    android:resizeMode="horizontal|vertical">
         //这个是Android3.0的一个新特性,是可以让widget改变大小,在2.3时候创建出来的Widget多大就是多大,不能改变,可以把这个去掉
     </appwidget-provider>
    
  4. 更新Widget数据的服务

     public class UpdateWidgetService extends Service {
         private Timer timer;
         private TimerTask task;
         @Override
         public IBinder onBind(Intent intent) {
             return null;
         }
         @Override
         public void onCreate() {
             super.onCreate();
             // 开启定期的任务更新widget.
             timer = new Timer();
             task = new TimerTask() {
                 @Override
                 public void run() {
                     AppWidgetManager awm = AppWidgetManager
                             .getInstance(getApplicationContext());
                     ComponentName component = new ComponentName(
                             getApplicationContext(), MyWidget.class);
                     RemoteViews views = new RemoteViews(getPackageName(),
                             R.layout.process_widget);
                     views.setTextViewText(
                             R.id.process_count,
                             "正在运行:"
                                     + ProcessStatusUtils.getProcessCount(getApplicationContext())
                                     + "个");
                     views.setTextViewText(
                             R.id.process_memory,
                             "可用内存:"
                                     + Formatter
                                             .formatFileSize(
                                                     getApplicationContext(), ProcessStatusUtils.getAvailRAM(getApplicationContext())));
                     Intent intent = new Intent();
                     intent.setAction("com.itheima.killall");
                     //设置一个自定义的广播事件 动作  com.itheima.killall
                     PendingIntent pendingIntent = PendingIntent.getBroadcast(getApplicationContext(), 0, intent, 0);
                     views.setOnClickPendingIntent(R.id.btn_clear, pendingIntent);
                     awm.updateAppWidget(component, views);
                 }
             };
             timer.schedule(task, 1000, 2000);
         }
         @Override
         public void onDestroy() {
             timer.cancel();
             task.cancel();
             timer = null;
             task = null;
             super.onDestroy();
         }
     }
    

Widget的声明周期

Widget就是一个特殊的广播接收者

  1. 当界面上第一个widget被创建的时候

     01-14 02:17:14.348: INFO/System.out(1853): onEnabled   当`widget`第一次被创建的时候调用. 非常适合做应用程序的初始化.
     01-14 02:17:14.348: INFO/System.out(1853): onReceive
     01-14 02:17:14.357: INFO/System.out(1853): onUpdate     当有新的`widget`被创建的时候 更新界面的操作. 当时间片到的时候`onupdate()`调用.
     01-14 02:17:14.357: INFO/System.out(1853): onReceive
    
  2. 当界面上第二个widget被创建的时候

     01-14 02:18:10.148: INFO/System.out(1853): onUpdate
     01-14 02:18:10.148: INFO/System.out(1853): onReceive
    
  3. 再创建新的widget
     01-14 02:18:10.148: INFO/System.out(1853): onUpdate
     01-14 02:18:10.148: INFO/System.out(1853): onReceive
    
  4. 从界面上移除一个widget
     01-14 02:19:11.709: INFO/System.out(1853): onDeleted
     01-14 02:19:11.709: INFO/System.out(1853): onReceive
    
  5. 最后一个widget被移除
     01-14 02:19:37.509: INFO/System.out(1853): onDeleted
     01-14 02:19:37.509: INFO/System.out(1853): onReceive
     01-14 02:19:37.509: INFO/System.out(1853): onDisabled  当`widget`从界面上全部移除的时候调用的方法. 非常适合删除临时文件停止后台服务.
     01-14 02:19:37.509: INFO/System.out(1853): onReceive
    
  6. widget就是一个特殊的广播接受者 当有新的事件产生的是 肯定会调用 onReceive();

注意: 在不同的手机上 widget的生命周期调用方法 可能有细微的不同. 360桌面 go桌面 awt桌面 腾讯桌面 小米桌面


results matching ""

    No results matching ""